home *** CD-ROM | disk | FTP | other *** search
/ Aminet 2 / Aminet AMIGA CDROM (1994)(Walnut Creek)[Feb 1994][W.O. 44790-1].iso / Aminet / util / gnu / apatch20.lha / patch-2.0.12u8 / NOT-NEEDED / malloc.c < prev    next >
C/C++ Source or Header  |  1992-09-16  |  15KB  |  470 lines

  1. /*
  2.  * @(#)nmalloc.c 1 (Caltech) 2/21/82
  3.  *
  4.  *      U of M Modified: 20 Jun 1983 ACT: strange hacks for Emacs
  5.  *
  6.  *      Nov 1983, Mike@BRL, Added support for 4.1C/4.2 BSD.
  7.  *
  8.  * This is a very fast storage allocator.  It allocates blocks of a small
  9.  * number of different sizes, and keeps free lists of each size.  Blocks
  10.  * that don't exactly fit are passed up to the next larger size.  In this
  11.  * implementation, the available sizes are (2^n)-4 (or -16) bytes long.
  12.  * This is designed for use in a program that uses vast quantities of
  13.  * memory, but bombs when it runs out.  To make it a little better, it
  14.  * warns the user when he starts to get near the end.
  15.  *
  16.  * June 84, ACT: modified rcheck code to check the range given to malloc,
  17.  * rather than the range determined by the 2-power used.
  18.  *
  19.  * Jan 85, RMS: calls malloc_warning to issue warning on nearly full.
  20.  * No longer Emacs-specific; can serve as all-purpose malloc for GNU.
  21.  * You should call malloc_init to reinitialize after loading dumped Emacs.
  22.  * Call malloc_stats to get info on memory stats if MSTATS turned on.
  23.  * realloc knows how to return same block given, just changing its size,
  24.  * if the power of 2 is correct.
  25.  */
  26.  
  27. /*
  28.  * nextf[i] is the pointer to the next free block of size 2^(i+3).  The
  29.  * smallest allocatable block is 8 bytes.  The overhead information will
  30.  * go in the first int of the block, and the returned pointer will point
  31.  * to the second.
  32.  *
  33. #ifdef MSTATS
  34.  * nmalloc[i] is the difference between the number of mallocs and frees
  35.  * for a given block size.
  36.  *
  37.  * #endif * MSTATS *
  38.  *
  39.  */
  40.  
  41. #define ISALLOC ((char) 0xf7)   /* magic byte that implies allocation */
  42. #define ISFREE ((char) 0x54)    /* magic byte that implies free block */
  43.                                 /* this is for error checking only */
  44.  
  45. extern char etext;
  46.  
  47. /* end of the program; can be changed by calling init_malloc */
  48. static char *endofpure = &etext;
  49.  
  50. #ifdef MSTATS
  51. static int nmalloc[30];
  52. static int nmal, nfre;
  53. #endif /* MSTATS */
  54.  
  55. /* If range checking is not turned on, all we have is a flag indicating
  56.    whether memory is allocated, an index in nextf[], and a size field; to
  57.    realloc() memory we copy either size bytes or 1<<(index+3) bytes depending
  58.    on whether the former can hold the exact size (given the value of
  59.    'index').  If range checking is on, we always need to know how much space
  60.    is allocated, so the 'size' field is never used. */
  61.  
  62. struct mhead {
  63.         char     mh_alloc;      /* ISALLOC or ISFREE */
  64.         char     mh_index;      /* index in nextf[] */
  65. /* Remainder are valid only when block is allocated */
  66.         unsigned short mh_size; /* size, if < 0x10000 */
  67. #ifdef rcheck
  68.         unsigned mh_nbytes;     /* number of bytes allocated */
  69.         int      mh_magic4;     /* should be == MAGIC4 */
  70. #endif /* rcheck */
  71.         };
  72.  
  73. /* Access free-list pointer of a block.
  74.   It is stored at block + 4.
  75.   This is not a field in the mhead structure
  76.   because we want sizeof (struct mhead)
  77.   to describe the overhead for when the block is in use,
  78.   and we do not want the free-list pointer to count in that.  */
  79.  
  80. #define CHAIN(a) \
  81.   (*(struct mhead **) (sizeof (char *) + (char *) (a)))
  82.  
  83. #ifdef rcheck
  84.  
  85. /* To implement range checking, we write magic values in at the beginning and
  86.    end of each allocated block, and make sure they are undisturbed whenever a
  87.    free or a realloc occurs. */
  88. /* Written in each of the 4 bytes following the block's real space */
  89. #define MAGIC1 0x55
  90. /* Written in the 4 bytes before the block's real space */
  91. #define MAGIC4 0x55555555
  92. #define ASSERT(p) if (!(p)) botch("p"); else
  93. static
  94. botch(s)
  95.         char *s;
  96. {
  97.  
  98.         printf("assertion botched: %s\n", s);
  99.         abort();
  100. }
  101. #define EXTRA  4                /* 4 bytes extra for MAGIC1s */
  102. #else
  103. #define ASSERT(p)
  104. #define EXTRA  0
  105. #endif /* rcheck */
  106.  
  107. /* nextf[i] is free list of blocks of size 2**(i + 3)  */
  108.  
  109. static struct mhead *nextf[30];
  110.  
  111. #ifdef  M_WARN
  112. /* Number of bytes of writable memory we can expect to be able to get */
  113. static int  lim_data;
  114. /* Level number of warnings already issued.
  115.   0 -- no warnings issued.
  116.   1 -- 75% warning already issued.
  117.   2 -- 85% warning already issued.
  118. */
  119. static int  warnlevel;
  120. #endif /* M_WARN */
  121.  
  122. /* nonzero once initial bunch of free blocks made */
  123. static int gotpool;
  124.  
  125. /* Cause reinitialization based on job parameters;
  126.   also declare where the end of pure storage is. */
  127. malloc_init (end)
  128.     char *end; {
  129.         endofpure = end;
  130. #ifdef  M_WARN
  131.         lim_data = 0;
  132.         warnlevel = 0;
  133. #endif /* M_WARN */
  134.         }
  135.  
  136. static
  137. morecore (nu)                   /* ask system for more memory */
  138.     register int nu; {          /* size index to get more of  */
  139.         char   *sbrk ();
  140.         register char  *cp;
  141.         register int    nblks;
  142.         register int    siz;
  143.  
  144. #ifdef  M_WARN
  145. #ifndef BSD42
  146. #ifdef USG
  147.         extern long ulimit ();
  148.         if (lim_data == 0)              /* find out how much we can get */
  149.             lim_data = ulimit (3, 0) - TEXT_START;
  150. #else   /*HMS: was endif */
  151.         if (lim_data == 0)              /* find out how much we can get */
  152.             lim_data = vlimit (LIM_DATA, -1);
  153. #endif /* USG */        /HMS:* was not here */
  154. #else
  155.         if (lim_data == 0) {
  156.                 struct rlimit   XXrlimit;
  157.  
  158.                 getrlimit (RLIMIT_DATA, &XXrlimit);
  159.                 lim_data = XXrlimit.rlim_cur;}  /* soft limit */
  160. #endif /* BSD42 */
  161. #endif /* M_WARN */
  162.  
  163.         /* On initial startup, get two blocks of each size up to 1k bytes */
  164.         if (!gotpool)
  165.             getpool (), getpool (), gotpool = 1;
  166.  
  167.         /* Find current end of memory and issue warning if getting near max */
  168.  
  169.         cp = sbrk (0);
  170.         siz = cp - endofpure;
  171. #ifdef  M_WARN
  172.         switch (warnlevel) {
  173.             case 0:
  174.                 if (siz > (lim_data / 4) * 3) {
  175.                         warnlevel++;
  176.                         malloc_warning ("Warning: past 75% of memory limit");}
  177.                 break;
  178.             case 1:
  179.                 if (siz > (lim_data / 20) * 17) {
  180.                         warnlevel++;
  181.                         malloc_warning ("Warning: past 85% of memory limit");}
  182.                 break;
  183.             case 2:
  184.                 if (siz > (lim_data / 20) * 19) {
  185.                         warnlevel++;
  186.                         malloc_warning ("Warning: past 95% of memory limit");}
  187.                 break;}
  188. #endif /* M_WARN */
  189.  
  190.         if ((int) cp & 0x3ff)   /* land on 1K boundaries */
  191.             sbrk (1024 - ((int) cp & 0x3ff));
  192.  
  193.         /* Take at least 2k, and figure out how many blocks of the desired size we're about to get */
  194.         nblks = 1;
  195.         if ((siz = nu) < 8)
  196.             nblks = 1 << ((siz = 8) - nu);
  197.  
  198.         if ((cp = sbrk (1 << (siz + 3))) == (char *) -1)
  199.             return;                     /* no more room! */
  200.         if ((int) cp & 7) {             /* shouldn't happen, but just in case */
  201.                 cp = (char *) (((int) cp + 8) & ~7);
  202.                 nblks--;}
  203.  
  204.         /* save new header and link the nblks blocks together */
  205.         nextf[nu] = (struct mhead *) cp;
  206.         siz = 1 << (nu + 3);
  207.         while (1) {
  208.                 ((struct mhead *) cp) -> mh_alloc = ISFREE;
  209.                 ((struct mhead *) cp) -> mh_index = nu;
  210.                 if (--nblks <= 0) break;
  211.                 CHAIN ((struct mhead *) cp) = (struct mhead *) (cp + siz);
  212.                 cp += siz;}
  213. /*      CHAIN ((struct mhead *) cp) = 0;        /* since sbrk() returns cleared core, this is already set */
  214.         }
  215.  
  216. static
  217. getpool () {
  218.         register int nu;
  219.         register char *cp = sbrk (0);
  220.  
  221.         if ((int) cp & 0x3ff)   /* land on 1K boundaries */
  222.             sbrk (1024 - ((int) cp & 0x3ff));
  223.  
  224.         /* Get 2k of storage */
  225.  
  226.         cp = sbrk (04000);
  227.         if (cp == (char *) -1)
  228.             return;
  229.  
  230.         /* Divide it into an initial 8-word block
  231.         plus one block of size 2**nu for nu = 3 ... 10.  */
  232.  
  233.         CHAIN (cp) = nextf[0];
  234.         nextf[0] = (struct mhead *) cp;
  235.         ((struct mhead *) cp) -> mh_alloc = ISFREE;
  236.         ((struct mhead *) cp) -> mh_index = 0;
  237.         cp += 8;
  238.  
  239.         for (nu = 0; nu < 7; nu++) {
  240.                 CHAIN (cp) = nextf[nu];
  241.                 nextf[nu] = (struct mhead *) cp;
  242.                 ((struct mhead *) cp) -> mh_alloc = ISFREE;
  243.                 ((struct mhead *) cp) -> mh_index = nu;
  244.                 cp += 8 << nu;}}
  245.  
  246. char *
  247. malloc (n)              /* get a block */
  248.     unsigned n; {
  249.         register struct  mhead *p;
  250.         register unsigned int  nbytes;
  251.         register int    nunits = 0;
  252.  
  253.         /* Figure out how many bytes are required, rounding up to the nearest
  254.         multiple of 4, then figure out which nextf[] area to use */
  255.         nbytes = (n + sizeof *p + EXTRA + 3) & ~3;
  256.                 {
  257.                 register unsigned int   shiftr = (nbytes - 1) >> 2;
  258.  
  259.                 while (shiftr >>= 1)
  260.                     nunits++;
  261.                 }
  262.  
  263.         /* If there are no blocks of the appropriate size, go get some */
  264.         /* COULD SPLIT UP A LARGER BLOCK HERE ... ACT */
  265.         if (nextf[nunits] == 0)
  266.             morecore (nunits);
  267.  
  268.         /* Get one block off the list, and set the new list head */
  269.         if ((p = nextf[nunits]) == 0)
  270.             return 0;
  271.         nextf[nunits] = CHAIN (p);
  272.  
  273.         /* Check for free block clobbered */
  274.         /* If not for this check, we would gobble a clobbered free chain ptr */
  275.         /* and bomb out on the NEXT allocate of this size block */
  276.         if (p -> mh_alloc != ISFREE || p -> mh_index != nunits)
  277. #ifdef rcheck
  278.             botch ("block on free list clobbered");
  279. #else
  280.             abort ();
  281. #endif /* rcheck */
  282.  
  283.         /* Fill in the info, and if range checking, set up the magic numbers */
  284.         p -> mh_alloc = ISALLOC;
  285. #ifdef rcheck
  286.         p -> mh_nbytes = n;
  287.         p -> mh_magic4 = MAGIC4;
  288.                 {
  289.                 register char  *m = (char *) (p + 1) + n;
  290.  
  291.                 *m++ = MAGIC1, *m++ = MAGIC1, *m++ = MAGIC1, *m = MAGIC1;
  292.                 }
  293. #else
  294.         p -> mh_size = n;
  295. #endif /* rcheck */
  296. #ifdef MSTATS
  297.         nmalloc[nunits]++;
  298.         nmal++;
  299. #endif /* MSTATS */
  300.         return (char *) (p + 1);}
  301.  
  302. free (mem)
  303.     char *mem; {
  304.         register struct mhead *p;
  305.                 {
  306.                 register char *ap = mem;
  307.  
  308.                 ASSERT (ap != 0);
  309.                 p = (struct mhead *) ap - 1;
  310.                 ASSERT (p -> mh_alloc == ISALLOC);
  311. #ifdef rcheck
  312.                 ASSERT (p -> mh_magic4 == MAGIC4);
  313.                 ap += p -> mh_nbytes;
  314.                 ASSERT (*ap++ == MAGIC1); ASSERT (*ap++ == MAGIC1);
  315.                 ASSERT (*ap++ == MAGIC1); ASSERT (*ap   == MAGIC1);
  316. #endif /* rcheck */
  317.                 }
  318.                 {
  319.                 register int nunits = p -> mh_index;
  320.  
  321.                 ASSERT (nunits <= 29);
  322.                 p -> mh_alloc = ISFREE;
  323.                 CHAIN (p) = nextf[nunits];
  324.                 nextf[nunits] = p;
  325. #ifdef MSTATS
  326.                 nmalloc[nunits]--;
  327.                 nfre++;
  328. #endif /* MSTATS */
  329.                 }
  330.         }
  331.  
  332. char *
  333. realloc (mem, n)
  334.     char *mem;
  335.     register unsigned n; {
  336.         register struct mhead *p;
  337.         register unsigned int tocopy;
  338.         register int nbytes;
  339.         register int nunits;
  340.  
  341.         if ((p = (struct mhead *) mem) == 0)
  342.             return malloc (n);
  343.         p--;
  344.         nunits = p -> mh_index;
  345.         ASSERT (p -> mh_alloc == ISALLOC);
  346. #ifdef rcheck
  347.         ASSERT (p -> mh_magic4 == MAGIC4);
  348.                 {
  349.                 register char *m = mem + (tocopy = p -> mh_nbytes);
  350.                 ASSERT (*m++ == MAGIC1); ASSERT (*m++ == MAGIC1);
  351.                 ASSERT (*m++ == MAGIC1); ASSERT (*m   == MAGIC1);
  352.                 }
  353. #else
  354.         if (p -> mh_index >= 13)
  355.             tocopy = (1 << (p -> mh_index + 3)) - sizeof *p;
  356.         else
  357.             tocopy = p -> mh_size;
  358. #endif /* rcheck */
  359.  
  360.         /* See if desired size rounds to same power of 2 as actual size. */
  361.         nbytes = (n + sizeof *p + EXTRA + 7) & ~7;
  362.  
  363.         /* If ok, use the same block, just marking its size as changed.  */
  364.         if (nbytes > (4 << nunits) && nbytes <= (8 << nunits)) {
  365. #ifdef rcheck
  366.                 register char *m = mem + tocopy;
  367.                 *m++ = 0;  *m++ = 0;  *m++ = 0;  *m++ = 0;
  368.                 p-> mh_nbytes = n;
  369.                 m = mem + n;
  370.                 *m++ = MAGIC1;  *m++ = MAGIC1;  *m++ = MAGIC1;  *m++ = MAGIC1;
  371. #else
  372.                 p -> mh_size = n;
  373. #endif /* rcheck */
  374.                 return mem;}
  375.  
  376.         if (n < tocopy)
  377.             tocopy = n;
  378.                 {
  379.                 register char *new;
  380.                 void bcopy();   /*HMS: here? */
  381.  
  382.                 if ((new = malloc (n)) == 0)
  383.                     return 0;
  384.                 bcopy (mem, new, tocopy);
  385.                 free (mem);
  386.                 return new;
  387.                 }
  388.         }
  389.  
  390. #ifdef MSTATS
  391. /* Return statistics describing allocation of blocks of size 2**n. */
  392.  
  393. struct mstats_value {
  394.         int blocksize;
  395.         int nfree;
  396.         int nused;
  397.         };
  398.  
  399. struct mstats_value
  400. malloc_stats (size)
  401.     int size; {
  402.         struct mstats_value v;
  403.         register int i;
  404.         register struct mhead *p;
  405.  
  406.         v.nfree = 0;
  407.  
  408.         if (size < 0 || size >= 30) {
  409.                 v.blocksize = 0;
  410.                 v.nused = 0;
  411.                 return v;}
  412.  
  413.         v.blocksize = 1 << (size + 3);
  414.         v.nused = nmalloc[size];
  415.  
  416.         for (p = nextf[size]; p; p = CHAIN (p))
  417.             v.nfree++;
  418.  
  419.         return v;}
  420. #endif
  421.  
  422. /* how much space is available? */
  423.  
  424. unsigned freespace() {
  425.         register int i, j;
  426.         register struct mhead *p;
  427.         register unsigned space = 0;
  428.         int local;      /* address only is used */
  429.  
  430.         space = (char *)&local - sbrk(0);       /* stack space */
  431.  
  432.         for (i = 0; i < 30; i++) {
  433.                 for (j = 0, p = nextf[i]; p; p = CHAIN (p), j++) ;
  434.                 space += j * (1 << (i + 3));}
  435.  
  436.         return(space);}
  437.  
  438. /* How big is this cell? */
  439.  
  440. unsigned mc_size(cp)
  441.     char *cp;{
  442.         register struct mhead *p;
  443.  
  444.         if ((p = (struct mhead *) cp) == 0) {
  445.                 /*HMS? */
  446.                 }
  447.         p--;
  448. #ifdef rcheck
  449.         return p -> mh_nbytes;
  450. #else
  451.         return (1 << (p -> mh_index + 3)) - sizeof *p;
  452. /**/
  453. /*      if (p -> mh_index >= 13)
  454. /*          return (1 << (p -> mh_index + 3)) - sizeof *p;
  455. /*      else
  456. /*          return p -> mh_size;
  457. /**/
  458. #endif /* rcheck */
  459.         }
  460.  
  461. /*HMS: Really should use memcpy, if available... */
  462.  
  463. void bcopy(source, dest, len)
  464.     register char *source, *dest;
  465.     register len; {
  466.         register i;
  467.  
  468.         for (i = 0; i < len; i++)
  469.             *dest++ = *source++;}
  470.